package com.atguigu.gulimall.search.thread;


import ch.qos.logback.classic.spi.IThrowableProxy;

import java.util.concurrent.*;

/**
 * @author liuzemin
 * @date 2021/2/2 12:33
 * @description
 */
public class ThreadTest {
    public static ExecutorService executor = Executors.newFixedThreadPool(10);

    public static void main(String[] args) throws ExecutionException, InterruptedException {
        System.out.println("main...start...");
//        CompletableFuture<Void> future = CompletableFuture.runAsync(() -> {
//            System.out.println("当前线程：" + Thread.currentThread().getId());
//            int i = 10 / 2;
//            System.out.println("运行结果：" + i);
//        }, executor);

        /**
         * 方法执行完成后的感知
         */
//        CompletableFuture<Integer> future = CompletableFuture.supplyAsync(() -> {
//            System.out.println("当前线程：" + Thread.currentThread().getId());
//            int i = 10 / 0;
//            System.out.println("运行结果：" + i);
//            return i;
//        }, executor).whenComplete((res,exception) -> {
//            //虽然能得到异常信息，但是没法修改返回数据
//            System.out.println("异步任务成功完成了。。。结果是："+res+";异常是："+exception);
//        }).exceptionally(throwable -> 10);//可以感知异常，同时返回默认值


        /**
         * 方法执行完成后的处理
         */
//        CompletableFuture<Integer> future = CompletableFuture.supplyAsync(() -> {
//            System.out.println("当前线程：" + Thread.currentThread().getId());
//            int i = 10 / 4;
//            System.out.println("运行结果：" + i);
//            return i;
//        }, executor).handle((res, throwable) -> {
//            if (res != null){
//                return res*2;
//            }
//            if (throwable!=null){
//                return 0;
//            }
//            return 0;
//        });

        /**
         * 线程串行化
         * 1、thenRun:不能获取到上一步的执行结果，无返回值
         *      .thenRunAsync(() -> {
         *             System.out.println("任务2启动了");
         *         }, executor);
         *
         * 2、thenAcceptAsync：能接受上一步结果，但是无返回值
         *      .thenAcceptAsync(res -> {
         *             System.out.println("任务2启动了"+res);
         *         },executor);
         *
         * 3、thenApplyAsync: 能接受上一步结果，有返回值
         *
         */
//        CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {
//            System.out.println("当前线程：" + Thread.currentThread().getId());
//            int i = 10 / 4;
//            System.out.println("运行结果：" + i);
//            return i;
//        }, executor).thenApplyAsync((res) -> {
//            System.out.println("任务2启动了。。。" + res);
//            return "hello" + res;
//        }, executor);

//        CompletableFuture<Object> future01 = CompletableFuture.supplyAsync(() -> {
//            System.out.println("任务1线程：" + Thread.currentThread().getId());
//            int i = 10 / 4;
//            System.out.println("任务1结果：" + i);
//            return i;
//        }, executor);
//
//        CompletableFuture<Object> future02 = CompletableFuture.supplyAsync(() -> {
//            System.out.println("任务2线程：" + Thread.currentThread().getId());
//            try {
//                Thread.sleep(3000);
//                System.out.println("任务2结果：" );
//            } catch (InterruptedException e) {
//                e.printStackTrace();
//            }
//            return "hello";
//        }, executor);

        /**
         * 两个都完成
         */
//        future01.runAfterBothAsync(future02,()->{
//            System.out.println("任务3开始。。");
//        },executor);
//        future01.thenAcceptBothAsync(future02,(f1,f2) -> {
//            System.out.println("任务3开始。。之前的结果："+f1+"-->"+f2);
//        },executor);
//        CompletableFuture<String> future = future01.thenCombineAsync(future02, (f1, f2) -> {
//            return f1 + ":" + f2 + "->  haha";
//        }, executor);

        /**
         * 两个任务，只要有一个完成，我们就执行任务3
         * runAfterEitherAsync: 不感知结果，没有返回值
         * acceptEitherAsync:  感知结果，没有返回结果
         */
//        future01.runAfterEitherAsync(future02,() -> {
//            System.out.println("任务3开始...之前的结果：");
//        },executor);

//        future01.acceptEitherAsync(future02,(res) -> {
//            System.out.println("任务3开始...之前的结果："+res);
//        },executor);

//        CompletableFuture<String> future = future01.applyToEitherAsync(future02, (res) -> {
//            System.out.println("任务3开始...之前的结果：" + res);
//            return res.toString() + "->haha";
//        }, executor);

        CompletableFuture<String> futureImg = CompletableFuture.supplyAsync(() -> {
            System.out.println("查询商品的图片信息");
            return "hello.jpg";
        },executor);

        CompletableFuture<String> futureAttr = CompletableFuture.supplyAsync(() -> {


                System.out.println("查询商品的属性");

            return "黑色+256G";
        },executor);

        CompletableFuture<String> futureDesc = CompletableFuture.supplyAsync(() -> {
            System.out.println("查询商品的介绍");
            return "华为";
        },executor);

//        CompletableFuture<Void> allOf = CompletableFuture.allOf(futureImg, futureAttr, futureDesc);
//        allOf.get();//等待所有结果完成
        CompletableFuture<Object> anyOf = CompletableFuture.anyOf(futureImg, futureAttr, futureDesc);


        System.out.println("main...end..."+anyOf.get());
    }

    public void thread(String[] args) throws ExecutionException, InterruptedException {
        System.out.println("main...start...");
        /**
         * 1、继承Thread
         *          Thread01 thread = new Thread01();
         *         thread.start();//启动线程
         * 2、实现Runnable接口
         *          Runnable01 runnable01 = new Runnable01();
         *         new Thread(runnable01).start();
         * 3、实现Callable接口 + FutureTask(可以拿到返回结果，可以处理异常)
         *          FutureTask<Integer> futureTask = new FutureTask<>(new Callable01());
         *         //阻塞等待整个线程执行完成，获取返回结果
         *         new Thread(futureTask).start();
         *         Integer integer = futureTask.get();
         * 4、线程池[ExecutorService]
         *          给线程池直接提交任务
         *          service.execute(new Runnable01());
         *          1、创建：
         *               1）、Executors
         *               2）、new ThreadPoolExecutor()
         *          Future：可以获取到异步结果
         *
         * 区别：
         *      1、2不能获得返回值。 3可以获取返回值
         *      1、2、3都不能控制资源
         *      4可以控制资源，性能稳定。
         */
        //我们以后在业务代码里面，以上三种启动线程的方式都不用。【将所有的多线程异步任务都交给线程池执行】
//        new Thread(()-> System.out.println("hello")).start();

        //当前系统中池只有一两个，每个异步任务，提交给线程池让他自己去执行就行
        /**
         * 七大参数
         * corePoolSize:[5]核心线程数(一直存在，除非（allowCoreThreadTimeOut）);线程池，创建好以后就准备就绪的线程数量，就等待来接受异步任务去执行.
         *          5个  Thread thread = new Thread();  thread.start();
         * maximumPoolSize:[200]最大线程数量；控制资源
         * keepAliveTime:存活时间。如果当前的线程数量大于core数量。
         *      释放空闲的线程(maximumPoolSize-corePoolSize)。只要线程空闲大于指定的keepAliveTime;
         * unit:时间单位
         * BlockingQueue<Runnable> workQueue:阻塞队列。如果任务有很多，就会将目前多的任务放在队列里面。
         *              只要有线程空闲，就会去队列里面去除新的任务继续执行。
         * threadFactory:线程的创建工厂。
         * RejectedExecutionHandler handler:如果队列满了，按照我们指定的拒绝策略拒绝执行任务
         *
         * 工作顺序：
         *      1、线程池创建，准备好core数量的核心线程，准备接受任务
         *          1.1、core满了，就将再进来的任务放入阻塞队列中。空闲的core就会自己去阻塞队列获取任务执行
         *          1.2、阻塞队列满了，就直接开新线程执行，最大只能开到max指定的数量
         *          1.3、max满了就用RejectedExecutionHandler handler拒绝任务
         *          1.4、max都执行完成，有很多空闲，再指定的时间keepAliveTime以后，释放max-core这些线程
         *
         *          new LinkedBlockingDeque<>():默认是Integer的最大值
         *
         */
        ThreadPoolExecutor executor = new ThreadPoolExecutor(5,
                200,
                10,
                TimeUnit.SECONDS,
                new LinkedBlockingDeque<>(100000),
                Executors.defaultThreadFactory(),
                new ThreadPoolExecutor.AbortPolicy());

//        Executors.newCachedThreadPool()     core是0，所有都可回收
//        Executors.newFixedThreadPool()      固定大小，core=max;都不可回收
//        Executors.newScheduledThreadPool()      定时任务的线程池
//        Executors.newSingleThreadExecutor()     单线程的线程池，后台从队列里面获取任务，挨个执行

        System.out.println("main...end...");
    }

    public static class Thread01 extends Thread{
        @Override
        public void run() {
            System.out.println("当前线程："+Thread.currentThread().getId());
            int i = 10/2;
            System.out.println("运行结果："+i);
        }
    }

    public static class Runnable01 implements Runnable{
        @Override
        public void run() {
            System.out.println("当前线程："+Thread.currentThread().getId());
            int i = 10/2;
            System.out.println("运行结果："+i);
        }
    }

    public static class Callable01 implements Callable<Integer> {
        @Override
        public Integer call() throws Exception {
            System.out.println("当前线程："+Thread.currentThread().getId());
            int i = 10/2;
            System.out.println("运行结果："+i);
            return i;
        }
    }
}
